home *** CD-ROM | disk | FTP | other *** search
/ Internet Info 1994 March / Internet Info CD-ROM (Walnut Creek) (March 1994).iso / networking / ip / ka9q / alpha.arc / NRS.C < prev    next >
C/C++ Source or Header  |  1988-07-26  |  6KB  |  259 lines

  1. /* This module implements the serial line framing method used by
  2.  * net/rom nodes.  This allows the net/rom software to talk to
  3.  * an actual net/rom over its serial interface, which is useful
  4.  * if we want to do packet switching for multi-line wormholes.
  5.  * Dan Frank, W9NK
  6.  */
  7. #include <stdio.h>
  8. #include "global.h"
  9. #include "mbuf.h"
  10. #include "iface.h"
  11. #include "ax25.h"
  12. #include "nrs.h"
  13. #include "asy.h"
  14. #include "trace.h"
  15.  
  16. int asy_output();
  17.  
  18. /* control structures, sort of overlayed on async control blocks */
  19. struct nrs nrs[ASY_MAX];
  20.  
  21. /* Send a raw net/rom serial frame */
  22. nrs_raw(interface,bp)
  23. struct interface *interface;
  24. struct mbuf *bp;
  25. {
  26.     dump(interface,IF_TRACE_OUT,TRACE_AX25,bp) ;
  27.     
  28.     /* Queue a frame on the output queue and start transmitter */
  29.     nrsq(interface->dev,bp);
  30. }
  31.  
  32. /* Encode a raw packet in net/rom framing, put on link output queue, and kick
  33.  * transmitter
  34.  */
  35. static
  36. nrsq(dev,bp)
  37. int16 dev;        /* Serial line number */
  38. struct mbuf *bp;    /* Buffer to be sent */
  39. {
  40.     register struct nrs *sp;
  41.     struct mbuf *nrs_encode();
  42.  
  43.     if((bp = nrs_encode(bp)) == NULLBUF)
  44.         return;    
  45.  
  46.     sp = &nrs[dev];
  47.     enqueue(&sp->sndq,bp);
  48.     sp->sndcnt++;
  49.     if(sp->tbp == NULLBUF)
  50.         nrasy_start(dev);
  51. }
  52.  
  53. /* Start output, if possible, on asynch device dev */
  54. static
  55. nrasy_start(dev)
  56. int16 dev;
  57. {
  58.     register struct nrs *sp;
  59.  
  60.     if(!stxrdy(dev))
  61.         return;        /* Transmitter not ready */
  62.  
  63.     sp = &nrs[dev];
  64.     if(sp->tbp != NULLBUF){
  65.         /* transmission just completed */
  66.         free_p(sp->tbp);
  67.         sp->tbp = NULLBUF;
  68.     }
  69.     if(sp->sndq == NULLBUF)
  70.         return;    /* No work */
  71.  
  72.     sp->tbp = dequeue(&sp->sndq);
  73.     sp->sndcnt--;
  74.     asy_output(dev,sp->tbp->data,sp->tbp->cnt);
  75. }
  76.  
  77. /* Encode a packet in net/rom serial format */
  78. static
  79. struct mbuf *
  80. nrs_encode(bp)
  81. struct mbuf *bp;
  82. {
  83.     struct mbuf *lbp;    /* Mbuf containing line-ready packet */
  84.     register char *cp;
  85.     char c;
  86.     unsigned char csum = 0 ;
  87.  
  88.     /* Allocate output mbuf that's twice as long as the packet.
  89.      * This is a worst-case guess (consider a packet full of STX's!)
  90.      * Add five bytes for STX, ETX, checksum, and two nulls.
  91.      */
  92.     lbp = alloc_mbuf(2*len_mbuf(bp) + 5);
  93.     if(lbp == NULLBUF){
  94.         /* No space; drop */
  95.         free_p(bp);
  96.         return NULLBUF;
  97.     }
  98.     cp = lbp->data;
  99.  
  100.     *cp++ = STX ;
  101.  
  102.     /* Copy input to output, escaping special characters */
  103.     while(pullup(&bp,&c,1) == 1){
  104.         switch(uchar(c)){
  105.         case STX:
  106.         case ETX:
  107.         case DLE:
  108.             *cp++ = DLE;
  109.             /* notice drop through to default */
  110.         default:
  111.             *cp++ = c;
  112.         }
  113.         csum += uchar(c) ;
  114.     }
  115.     *cp++ = ETX;
  116.     *cp++ = csum ;
  117.     *cp++ = NUL ;
  118.     *cp++ = NUL ;
  119.     
  120.     lbp->cnt = cp - lbp->data;
  121.     return lbp;
  122. }
  123. /* Process incoming bytes in net/rom serial format
  124.  * When a buffer is complete, return it; otherwise NULLBUF
  125.  */
  126. static
  127. struct mbuf *
  128. nrs_decode(dev,c)
  129. int16 dev;    /* net/rom unit number */
  130. char c;        /* Incoming character */
  131. {
  132.     struct mbuf *bp;
  133.     register struct nrs *sp;
  134.  
  135.     sp = &nrs[dev];
  136.     switch(sp->state) {
  137.         case NRS_INTER:
  138.             if (uchar(c) == STX) {    /* look for start of frame */
  139.                 sp->state = NRS_INPACK ;    /* we're in a packet */
  140.                 sp->csum = 0 ;                /* reset checksum */
  141.             }
  142.             return NULLBUF ;
  143.             break ;    /* just for yucks */
  144.         case NRS_CSUM:
  145.             bp = sp->rbp ;
  146.             sp->rbp = NULLBUF ;
  147.             sp->rcnt = 0 ;
  148.             sp->state = NRS_INTER ;    /* go back to inter-packet state */
  149.             if (sp->csum == uchar(c)) {
  150.                 sp->packets++ ;
  151.                 return bp ;
  152.             }
  153.             else {
  154.                 free_p(bp) ;    /* drop packet with bad checksum */
  155.                 sp->errors++ ;    /* increment error count */
  156.                 return NULLBUF ;
  157.             }
  158.             break ;
  159.         case NRS_ESCAPE:
  160.             sp->state = NRS_INPACK ;    /* end of escape */
  161.             break ;            /* this will drop through to char processing */
  162.         case NRS_INPACK:
  163.             switch (uchar(c)) {
  164.                 /* If we see an STX in a packet, assume that previous */
  165.                 /* packet was trashed, and start a new packet */
  166.                 case STX:
  167.                     free_p(sp->rbp) ;
  168.                     sp->rbp = NULLBUF ;
  169.                     sp->rcnt = 0 ;
  170.                     sp->csum = 0 ;
  171.                     sp->errors++ ;
  172.                     return NULLBUF ;
  173.                     break ;
  174.                 case ETX:
  175.                     sp->state = NRS_CSUM ;    /* look for checksum */
  176.                     return NULLBUF ;
  177.                     break ;
  178.                 case DLE:
  179.                     sp->state = NRS_ESCAPE ;
  180.                     return NULLBUF ;
  181.                     break ;
  182.             }
  183.     }
  184.     /* If we get to here, it's with a character that's part of the packet.
  185.      * Make sure there's space for it.
  186.      */
  187.     if(sp->rbp == NULLBUF){
  188.         /* Allocate first mbuf for new packet */
  189.         if((sp->rbp1 = sp->rbp = alloc_mbuf(NRS_ALLOC)) == NULLBUF) {
  190.             sp->state = NRS_INTER ;
  191.             return NULLBUF; /* No memory, drop */
  192.         }
  193.         sp->rcp = sp->rbp->data;
  194.     } else if(sp->rbp1->cnt == NRS_ALLOC){
  195.         /* Current mbuf is full; link in another */
  196.         if((sp->rbp1->next = alloc_mbuf(NRS_ALLOC)) == NULLBUF){
  197.             /* No memory, drop whole thing */
  198.             free_p(sp->rbp);
  199.             sp->rbp = NULLBUF;
  200.             sp->rcnt = 0;
  201.             sp->state = NRS_INTER ;
  202.             return NULLBUF;
  203.         }
  204.         sp->rbp1 = sp->rbp1->next;
  205.         sp->rcp = sp->rbp1->data;
  206.     }
  207.     /* Store the character, increment fragment and total
  208.      * byte counts
  209.      */
  210.     *sp->rcp++ = c;
  211.     sp->rbp1->cnt++;
  212.     sp->rcnt++;
  213.     sp->csum += uchar(c) ;    /* add to checksum */
  214.     return NULLBUF;
  215. }
  216.  
  217. /* Process net/rom serial line I/O */
  218. void
  219. nrs_recv(interface)
  220. struct interface *interface;
  221. {
  222.     char c;
  223.     struct mbuf *bp;
  224.     int16 dev;
  225.     int16 asy_recv();
  226.     int ax_recv() ;
  227.  
  228.     dev = interface->dev;
  229.     /* Process any pending input */
  230.     while(asy_recv(dev,&c,1) != 0)
  231.         if((bp = nrs_decode(dev,c)) != NULLBUF) {
  232.             dump(interface,IF_TRACE_IN,TRACE_AX25,bp) ;
  233.             ax_recv(interface,bp);
  234.         }
  235.  
  236.     /* Kick the transmitter if it's idle */
  237.     if(stxrdy(dev))
  238.         nrasy_start(dev);
  239. }
  240.  
  241. /* donrstat:  display status of active net/rom serial interfaces */
  242. donrstat(argc,argv)
  243. int argc ;
  244. char *argv[] ;
  245. {
  246.     register struct nrs *np ;
  247.     register int i ;
  248.  
  249.     printf("Interface  SndQ  RcvB  NumReceived  CSumErrors\n") ;
  250.  
  251.     for (i = 0, np = nrs ; i < ASY_MAX ; i++, np++)
  252.         if (np->iface != NULLIF)
  253.             printf(" %8s   %3d  %4d   %10lu  %10lu\n",
  254.                     np->iface->name, np->sndcnt, np->rcnt,
  255.                     np->packets, np->errors) ;
  256.  
  257.     return 0 ;
  258. }
  259.